home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1998 October / Macworld (1998-10).dmg / Shareware World / Info / For Developers / MacZoop 1.8.4 / Required Classes / Z Sources / ZApplication.cpp < prev    next >
Encoding:
C/C++ Source or Header  |  1998-07-20  |  49.7 KB  |  1,948 lines  |  [TEXT/CWIE]

  1. /*************************************************************************************************
  2. *
  3. *
  4. *            MacZoop - "the framework for the rest of us"         
  5. *
  6. *
  7. *
  8. *            ZApplication.cpp    -- the application object
  9. *
  10. *
  11. *
  12. *
  13. *
  14. *            © 1996, Graham Cox
  15. *
  16. *
  17. *
  18. *
  19. *************************************************************************************************/
  20.  
  21.  
  22. #include    "MacZoop.h"
  23. #include    "ProjectSettings.h"
  24. #include    "ZEventHandler.h"
  25. #include    "ZWindow.h"
  26. #include    "ZPrinter.h"
  27. #include    "ZUndoTask.h"
  28.  
  29. #if APPEARANCE_MGR_AWARE
  30. #include    <Appearance.h>
  31. #endif
  32.  
  33. #if _USE_NAVIGATION_SERVICES
  34. #include    <Navigation.h>
  35. #endif
  36.  
  37. #include    <StandardFile.h>
  38. #include    <Notification.h>
  39.  
  40. // gApplication is the global application object. There is only one, naturally.
  41.  
  42. ZApplication*    gApplication = NULL;
  43.  
  44. // globals inited by this class
  45.  
  46. ZMenuBar*        gMenuBar = NULL;    // the main menubar object
  47. OSType            gAppSignature;        // the application's signature, obtained from 'BNDL', etc.    
  48. tMacInfo        gMacInfo;            // common gestalt results
  49. RgnHandle        gUtilRgn = NULL;    // general purpose region, handy as temp variable. Do NOT dispose!
  50. ZPrefsFile*        gPrefsFile = NULL;    // NOT inited unless your application subclass does it!    
  51.  
  52. // static functions used by the application object
  53.  
  54. static pascal long     ZGrowFunc( Size bytesShort );
  55. static GrowZoneUPP    gGZFunc = NewGrowZoneProc( ZGrowFunc );
  56.  
  57. #if _USE_NAVIGATION_SERVICES
  58. static pascal void    ZNavEventCallback(     NavEventCallbackMessage cbMessage,
  59.                                         NavCBRecPtr    cbParams,
  60.                                         NavCallBackUserData    cbData );
  61. NavEventUPP        gNavEventHandler = NewNavEventProc( ZNavEventCallback );
  62. #endif
  63.  
  64. // note: ZApplication is NEVER instantiated from a stream- to ensure this but keep
  65. // special cases out of the class registry, etc, we jig the construction function
  66. // to simply return gApplication.
  67.  
  68. ZObject*     CF_ZApplication()
  69. {
  70.     return gApplication;
  71. }
  72.  
  73.  
  74. extern ZObject* CF_ZObjectList();
  75.  
  76. /*--------------------------------***  CONSTRUCTOR  ***---------------------------------*/
  77.  
  78. ZApplication::ZApplication()
  79.     : ZCommander( NULL )
  80. {
  81.     long    qdFeatures;
  82.     OSErr    theErr;
  83.     
  84.     classID = CLASS_ZApplication;
  85.     
  86.     gUtilRgn = NewRgn();
  87.     done = FALSE;
  88.     phase = kInitialising;
  89.     zEH = NULL;
  90.     curUndoTask = NULL;
  91.     itsPrinter = NULL;
  92.     
  93.     appResRefNum = CurResFile();
  94.     
  95.     // set initial filetypes list to a zero handle
  96.     
  97.     itsFileTypes = (FTypeListHdl) NewHandleClear( sizeof( FTypeList ) - sizeof( OSType ));
  98.         
  99.     shortageFund = NULL;
  100.     memIsShort = FALSE;
  101.     userHasSeenAlert = FALSE;
  102.     
  103.     // check to see if we are running on a colour Mac
  104.     
  105.     theErr = Gestalt( gestaltQuickdrawFeatures, &qdFeatures );
  106.     gMacInfo.supportsColour = ((theErr == noErr) && (qdFeatures & 1));
  107.     
  108.     // check for the drag manager
  109.     
  110.     theErr = Gestalt( gestaltDragMgrAttr, &qdFeatures );
  111.     gMacInfo.hasDragManager = ((theErr == noErr) && (qdFeatures & 1));
  112.     
  113.     // check for a FPU
  114.     
  115.     theErr = Gestalt( gestaltFPUType, &qdFeatures );
  116.     gMacInfo.hasFPU = ((theErr == noErr) && ((qdFeatures & 1) == 0));
  117.     
  118.     // check for applescript
  119.     
  120.     theErr = Gestalt( gestaltAppleEventsAttr, &qdFeatures );
  121.     gMacInfo.hasAppleEvents = ((theErr == noErr) && (qdFeatures & 1));
  122.     
  123.     // check for QuickTime™
  124.     
  125.     theErr = Gestalt( gestaltQuickTimeVersion, &qdFeatures );
  126.     gMacInfo.hasQuickTime = ((theErr == noErr) && (qdFeatures > 0));
  127.  
  128.     // check for ICM
  129.     
  130.     theErr = Gestalt( gestaltCompressionMgr, &qdFeatures );
  131.     gMacInfo.hasImgCompressionMgr = (theErr == noErr);
  132.  
  133.     // check for appearance manager
  134.     
  135.     #if APPEARANCE_MGR_AWARE
  136.         theErr = Gestalt( gestaltAppearanceAttr, &qdFeatures );
  137.         gMacInfo.hasAppearanceMgr = (( theErr == noErr ) && ( qdFeatures & 1 ));
  138.         
  139.         #if __powerc
  140.         if ( gMacInfo.hasAppearanceMgr )
  141.             gMacInfo.hasAppearanceMgr = ((long)  CreateRootControl != kUnresolvedCFragSymbolAddress );
  142.         #endif
  143.     
  144.     #else
  145.         gMacInfo.hasAppearanceMgr = FALSE;
  146.     
  147.     #endif
  148.     
  149.     // what is the system version?
  150.     
  151.     theErr = Gestalt( gestaltSystemVersion, &qdFeatures );
  152.     gMacInfo.systemVersion = LoWord( qdFeatures );
  153.     
  154.     // check for navigation services:
  155.     
  156.     #if _USE_NAVIGATION_SERVICES
  157.     if ( gMacInfo.systemVersion >= 0x0755 )
  158.         gMacInfo.hasNavigationServices = NavServicesAvailable();
  159.     #else
  160.         gMacInfo.hasNavigationServices = FALSE;
  161.     #endif
  162.         
  163.     // initialise the animating cursors, and set the
  164.     // animated watch cursor going.
  165.     
  166.     AppCursorInit();
  167.     gApplication = this;
  168. }
  169.  
  170. /*--------------------------------***  DESTRUCTOR  ***---------------------------------*/
  171.  
  172.  
  173. ZApplication::~ZApplication()
  174. {
  175.     if ( gPrefsFile )
  176.         ForgetObject( gPrefsFile );
  177.     
  178.     if ( zEH )
  179.         ForgetObject( zEH );
  180.         
  181.     if ( itsPrinter )
  182.         ForgetObject( itsPrinter );
  183.         
  184.     if ( curUndoTask )
  185.         ForgetObject( curUndoTask );
  186.         
  187.     if ( itsFileTypes )
  188.         DisposeHandle((Handle) itsFileTypes);
  189.         
  190.     ForgetObject( gMenuBar );
  191.     StopCursorAnimation();
  192. }
  193.  
  194.  
  195. /*--------------------------------***  INITMACZOOP  ***---------------------------------*/
  196. /*    
  197.  
  198. Initialises the application, makes the event handler object and the menubar
  199.  
  200. ----------------------------------------------------------------------------------------*/
  201.  
  202. void        ZApplication::InitMacZoop( const short numMasterBlocks )
  203. {
  204.     if ( phase == kInitialising )
  205.     {
  206.         // init the mac toolbox, memory mangler etc.
  207.         
  208.         RegisterClasses();
  209.         InitMacApplication( numMasterBlocks );
  210.         
  211.         // create the memory shortage fund
  212.         
  213.         FailNIL( shortageFund = NewHandle( kShortageFundSize ));
  214.         SetGrowZone( gGZFunc );
  215.         
  216.         SetWatchCursor();
  217.         
  218.         // make the event handler object
  219.         
  220.         MakeEventHandler();
  221.     
  222.         // make the clipboard object
  223.         
  224.         MakeClipboard();
  225.         
  226.         // read any prefs that the user may have set up (default does nothing). A typical
  227.         // thing to do here is to simply instantiate a ZPrefsFile object and assign it to
  228.         // <gPrefsFile>. You may also want to open/read the file! (use of data or resource
  229.         // fork is up to you, so ZPrefsFile does not automatically perform the open step).
  230.         
  231.         ReadPrefs();
  232.         
  233.         // initialise menubar and make printer object
  234.         
  235.         InitMenuBar();
  236.         MakePrinter();
  237.         
  238.         // preload the navigation services if required, to make use of these faster later on.
  239.         
  240.         #if _USE_NAVIGATION_SERVICES
  241.         if ( gMacInfo.hasNavigationServices )
  242.             FailOSErr( NavLoad());
  243.         #endif
  244.         
  245.         // call user-function as last part of initialisation- default
  246.         // does nothing but can be overridden to do further set up.
  247.         
  248.         StartUp();
  249.         
  250.         // update the menubar- note that this is not drawn until StartUp completes,
  251.         // since we allow the programmer the flexibility to use StartUp to create & install
  252.         // dynamic menus, etc. if they want.
  253.         
  254.         gMenuBar->UpdateMenuBar();
  255.         
  256.         // the cursor is not reset here- we let the main event loop do that so that if
  257.         // events are initially available, the cursor keeps right on animating until
  258.         // the user gets a chance to do anything.
  259.         
  260.         phase = kRunning;
  261.     }
  262. }
  263.  
  264. /*-----------------------------***  INITMACAPPLICATION  ***-----------------------------*/
  265. /*    
  266.  
  267. Initialises the mac toolbox and memory manager
  268.  
  269. ----------------------------------------------------------------------------------------*/
  270.  
  271.     
  272. void        ZApplication::InitMacApplication( const short numMasterBlocks )
  273. {
  274.     // chant the "Macintosh mantra"...
  275.     
  276.     InitGraf( &qd.thePort );
  277.     InitFonts();
  278.     InitWindows();
  279.     InitMenus();
  280.     TEInit();
  281.     InitDialogs( NULL );
  282.     
  283.     // clear out the event queue in case any stray clicks or keypresses are left there
  284.     
  285.     FlushEvents( everyEvent, 0 );
  286.  
  287.     // see if the program can run on this Mac- if not, we show an alert and exit straight away
  288.     
  289.     if (! CheckCanRun())
  290.     {
  291.         StopCursorAnimation();
  292.         (void) Alert( kCantRunAlertID, NULL );
  293.         ExitToShell();
  294.     }    
  295.     else
  296.     {
  297.         // give yourself enough memory. For a bigger app, you may need to call MoreMasters a few
  298.         // more times. The parameter to this method sets the number of times MoreMasters is
  299.         // called, defaulting to 8, which gives 8 x 64 = 512 handles.
  300.         
  301.         MaxApplZone();
  302.         
  303.         short n = numMasterBlocks;
  304.         
  305.         while( n-- )
  306.             MoreMasters();
  307.             
  308.         // if we want appearance, register the app:
  309.         
  310.     #if APPEARANCE_MGR_AWARE
  311.     
  312.         FailOSErr( RegisterAppearanceClient());
  313.     
  314.     #endif
  315.             
  316.         // set up <gAppSignature>, either from BNDL resource, or from constant according
  317.         // to the user's project settings.
  318.         
  319.     #if USE_SIGNATURE_FROM_BNDL
  320.         
  321.         OSType**    bndlHand = (OSType**) GetResource( 'BNDL', 128 );
  322.         
  323.         if ( bndlHand )
  324.         {
  325.             gAppSignature = **bndlHand;
  326.             ReleaseResource((Handle) bndlHand );
  327.             
  328.             #if CHECK_FREF_RESOURCE_TYPES
  329.             
  330.             // first look for an 'open' resource ID=128. If there is one, we use it in
  331.             // preference to the FREF's, since there is a slightly different meaning. An 'open'
  332.             // resource can be treated as a list of OSTypes if you ignore the first 8 bytes.
  333.             
  334.             bndlHand = (OSType**) GetResource( 'open', 128 );
  335.             
  336.             if ( bndlHand )
  337.             {
  338.                 // we have an 'open' resource, so just copy the data (ignoring first 8 bytes)
  339.                 // to the file types list. Warning: This technique does not check for duplicates,
  340.                 // either in the resource or in the types list.
  341.                 
  342.                 long    bc = GetHandleSize((Handle) bndlHand );
  343.                 
  344.                 SetHandleSize((Handle) itsFileTypes, bc );
  345.                 FailMemError();
  346.                 
  347.                 BlockMoveData((Ptr) *bndlHand, (Ptr) *itsFileTypes, bc );
  348.                 ReleaseResource((Handle) bndlHand );
  349.             }
  350.             else
  351.             {
  352.                 // scan the 'FREF' resources and build the <itsFileTypes> list. We do NOT add
  353.                 // types of 'APPL', 'cdev', 'INIT', or 'rdev'- if you want to display such
  354.                 // files in the Open dialog, you have to call AddFileType() for these types directly.
  355.                 // This is a feature to prevent weird behaviour in the typical case.
  356.                 
  357.                 // how many 'FREF's do we have?
  358.                 
  359.                 short    i, fc = Count1Resources( 'FREF' );
  360.                 
  361.                 for( i = 1; i <= fc; i++ )
  362.                 {
  363.                     bndlHand = (OSType**) GetIndResource( 'FREF', i );
  364.                 
  365.                     if ( bndlHand )
  366.                     {
  367.                         // check if this is one of our 'forbidden' types:
  368.                         
  369.                         if ( **bndlHand != 'APPL'    &&
  370.                              **bndlHand != 'APPC'    &&
  371.                              **bndlHand != 'APPD'    &&
  372.                              **bndlHand != 'cdev'    &&
  373.                              **bndlHand != 'rdev'    &&
  374.                              **bndlHand != 'INIT' )
  375.                             AddFileType( **bndlHand );
  376.                         
  377.                         ReleaseResource((Handle) bndlHand );
  378.                     }
  379.                 }
  380.             }
  381.             
  382.             #endif    
  383.         }
  384.         else
  385.             gAppSignature = kUnknownSignature;
  386.     #else
  387.         gAppSignature = kApplicationSignature;
  388.     #endif
  389.     
  390.         (*itsFileTypes)->appSignature = gAppSignature;
  391.     }
  392. }
  393.  
  394. /*--------------------------------***  CHECKCANRUN  ***---------------------------------*/
  395. /*    
  396.  
  397. Examine the machine environment to see if this will run on this Mac. By default, it can
  398. on System 7 or later.
  399.  
  400. ----------------------------------------------------------------------------------------*/
  401.  
  402. Boolean        ZApplication::CheckCanRun()
  403. {
  404.     // Returning FALSE will cause the program to
  405.     // immediately show an alert and quit.
  406.     
  407.     // by default, we can run on any Mac with System 7.0 or later.
  408.     
  409.     return( gMacInfo.systemVersion >= 0x0700 );    
  410. }
  411.  
  412. #pragma mark -
  413. /*----------------------------------***  GETNAME  ***-----------------------------------*/
  414. /*    
  415. Get the user-visible name of the application, as seen in the Finder. This works even if
  416. the user has renamed the app, since it finds the filename.
  417. ----------------------------------------------------------------------------------------*/
  418.  
  419.  
  420. void        ZApplication::GetName( Str255 appName )
  421. {
  422.     CopyPString( LMGetCurApName(), appName );
  423. }
  424.  
  425. /*------------------------------------***  RUN  ***-------------------------------------*/
  426. /*    
  427.  
  428. Fetch events and handle them until the user quits
  429.  
  430. ----------------------------------------------------------------------------------------*/
  431.  
  432. void        ZApplication::Run()
  433. {
  434.     // runs the show by asking the event object to get events and handle them. This goes on
  435.     // forever until done is set to TRUE.
  436.     
  437.     if ( phase == kRunning )
  438.     {
  439.         while(! done)
  440.         {
  441.             try
  442.             {
  443.                 // repeatedly get an event, handle an event
  444.                 
  445.                 Process1Event();
  446.                 
  447.                 // deal with the memory shortage situation, if one has arisen as
  448.                 // a result of the last event.
  449.                 
  450.                 CheckLowMemory();
  451.             }
  452.             catch( OSErr theErr )
  453.             {
  454.                 // if here, an exception was thrown. If the error was userCanceledErr,
  455.                 // the alert is not shown, since the exception is legitimate. Thus for
  456.                 // many operations, you need only throw a userCanceledErr exception to
  457.                 // simply invoke the cancel.
  458.                 
  459.                 StopCursorAnimation();
  460.                 HandleError( theErr );
  461.                 
  462.                 // the buck stops here- no exceptions will be thrown beyond this point.
  463.             }
  464.             // stop any animating cursors. This means that a lengthy process need
  465.             // only set the cursor going and can then forget about it. When the
  466.             // app resumes handling events, it will be automatically cancelled.
  467.             
  468.             StopCursorAnimation();
  469.         }
  470.         done = TRUE;
  471.         phase = kQuitting;
  472.     }
  473. }
  474.  
  475.  
  476. /*-----------------------------***  CHECKLOWMEMORY  ***---------------------------------*/
  477. /*    
  478. check if a low memory situation has arisen. If so, inform the user and manage the re-
  479. plenishment of the shortage fund.
  480. ----------------------------------------------------------------------------------------*/
  481.  
  482. void        ZApplication::CheckLowMemory()
  483. {
  484.     if ( memIsShort )
  485.     {
  486.         // some of the shortage fund was used. Try to replenish it:
  487.         
  488.         memIsShort = FALSE;
  489.         
  490.         Handle temp = NewHandle( kShortageFundSize );
  491.         
  492.         // if that resets memIsShort, then the grow zone func was called, so we
  493.         // are not in the clear yet. However, if the grow zone func wasn't
  494.         // called, then we can safely get rid of that handle and replace it with
  495.         // this one
  496.         
  497.         if ( memIsShort )
  498.         {
  499.             if ( temp )
  500.                 DisposeHandle( temp );
  501.             
  502.             // couldn't replenish the fund, so if the user hasn't seen the
  503.             // warning yet, show it now.
  504.             
  505.             if (! userHasSeenAlert)
  506.             {
  507.                 StopCursorAnimation();
  508.                 (void) NotifyAlert( kMemoryLowAlertID );
  509.                 userHasSeenAlert = TRUE;
  510.             }
  511.             
  512.             // if memIsShort is TRUE, it affects UpdateMenus such that New and Open are
  513.             // greyed out. This is an attempt to stop the user creating things that will
  514.             // eat up even more memory. You might want to use the same technique for commands
  515.             // of your own that may allocate lots of memory. For this reason, memIsShort is
  516.             // a public member.
  517.         }
  518.         else
  519.         {
  520.             // fund replenished, so get rid of any remaining fund and replace it
  521.             // with the newly allocated shortage fund.
  522.             
  523.             DisposeHandle( shortageFund );
  524.             shortageFund = temp;
  525.             userHasSeenAlert = FALSE;
  526.         }
  527.     }
  528. }
  529.  
  530.  
  531. /*------------------------------***  MEMORYSHORTAGE  ***--------------------------------*/
  532. /*    
  533.  
  534. called from the growzone proc when memory needs to be freed. This releases some or all
  535. of the shortage fund, but you can override it to make additional sacrifices if need be.
  536. ----------------------------------------------------------------------------------------*/
  537.  
  538. Boolean        ZApplication::MemoryShortage( const Size bytesShort )
  539. {
  540.     // this is called when the memory manager gets into dire straits. We can free some or all
  541.     // of our emergency fund to satisfy the request. If we succeed, we return TRUE, else FALSE.
  542.     
  543.     Size    fundSize = 0;
  544.     Handle    gzHandle;
  545.     
  546.     // get the handle the mem manager is dealing with at the moment. This is important since
  547.     // this might be the shortage fund itself. If it is, we can't resize it, so we really
  548.     // are in deep do-do. In this case, we flag memIsShort and hope the user will not ignore
  549.     // the message!
  550.     
  551.     gzHandle = GZSaveHnd();
  552.     
  553.     if ( gzHandle != shortageFund )
  554.     {
  555.         fundSize = GetHandleSize( shortageFund );
  556.         
  557.         // release all or some of the memory to try and satisfy the request
  558.         
  559.         if ( fundSize <= bytesShort )
  560.             SetHandleSize( shortageFund, 0 );
  561.         else
  562.             SetHandleSize( shortageFund, fundSize - bytesShort );
  563.     }
  564.     // flag the shortage so we can inform the user
  565.     
  566.     memIsShort = TRUE;
  567.     
  568.     // did we actually manage to free the requested amount?
  569.     
  570.     return( fundSize > bytesShort );
  571. }
  572.  
  573.  
  574. /*------------------------------***  PROCESS1EVENT  ***---------------------------------*/
  575. /*    
  576. get an event, dispatch an event, get an event, dispatch...
  577. ----------------------------------------------------------------------------------------*/
  578.  
  579. void        ZApplication::Process1Event()
  580. {
  581.     EventRecord        theEvent;
  582.     
  583.     zEH->GetAnEvent( &theEvent );
  584.     zEH->DispatchAnEvent( &theEvent );
  585. }
  586.  
  587.  
  588. /*------------------------------***  PROCESS1EVENT  ***---------------------------------*/
  589. /*    
  590. dispatch an event that was fetched externally, for example by a library such as Nav
  591. Services.
  592. ----------------------------------------------------------------------------------------*/
  593.  
  594. void        ZApplication::Process1Event( EventRecord* anExternalEvent )
  595. {
  596.     zEH->DispatchAnEvent( anExternalEvent );
  597. }
  598.  
  599.  
  600. /*-----------------------------***  GETCURRENTEVENT  ***--------------------------------*/
  601. /*
  602. return the current event in progress, with FALSE if it was a null event, else TRUE.    
  603. ----------------------------------------------------------------------------------------*/
  604.  
  605. Boolean        ZApplication::GetCurrentEvent( EventRecord* anEvent )
  606. {
  607.     zEH->GetLatestEvent( anEvent );
  608.     
  609.     return( anEvent->what != nullEvent );
  610. }
  611.  
  612.  
  613. /*--------------------------------***  GETCLICKS  ***-----------------------------------*/
  614. /*    
  615. return the number of clicks counted by the event handler in the same place. Returns 1 for
  616. single click, 2 for double, 3 for triple, etc.
  617. ----------------------------------------------------------------------------------------*/
  618.  
  619. short        ZApplication::GetClicks()
  620. {
  621.     return zEH->GetClicks();
  622. }
  623.  
  624.  
  625. /*-------------------------------***  INBACKGROUND  ***---------------------------------*/
  626. /*    
  627. return TRUE if the application is in the background, FALSE in foreground.
  628. ----------------------------------------------------------------------------------------*/
  629.  
  630. Boolean        ZApplication::InBackground()
  631. {
  632.     return zEH->InBackground();
  633. }
  634.  
  635.  
  636. /*-----------------------------------***  QUIT  ***-------------------------------------*/
  637. /*    
  638.  
  639. close all of the windows. If successful, delete the application and return TRUE
  640.  
  641. ----------------------------------------------------------------------------------------*/
  642.  
  643. Boolean        ZApplication::Quit()
  644. {
  645.     if ( phase == kQuitting )
  646.     {
  647.         CloseAll();
  648.         
  649.         // the Quit can be abandoned by resetting <done> to FALSE. If this
  650.         // has not occurred, then truly say goodbye.
  651.         
  652.         if ( done )
  653.         {
  654.             #if _USE_NAVIGATION_SERVICES
  655.             if ( gMacInfo.hasNavigationServices )
  656.                 FailOSErr( NavUnload());
  657.             #endif
  658.             
  659.             #if APPEARANCE_MGR_AWARE
  660.             
  661.             (void) UnregisterAppearanceClient();
  662.             
  663.             #endif
  664.             
  665.             ShutDown();
  666.             ForgetThis();
  667.         }
  668.     }
  669.     return done;
  670. }
  671.  
  672.  
  673. /*-------------------------------***  REQUESTQUIT  ***----------------------------------*/
  674. /*    
  675.  
  676. ask the application to quit. This is called by choosing Quit from the File menu, for example.
  677.  
  678. ----------------------------------------------------------------------------------------*/
  679.  
  680.  
  681. void        ZApplication::RequestQuit()
  682. {
  683.     done = TRUE;
  684. }
  685.  
  686. /*----------------------------***  HANDLEAPPLEEVENT  ***--------------------------------*/
  687. /*    
  688. Handle the four required apple events. If you override this to handle your own apple
  689. events, be sure to call the inherited method for everything else.
  690. ----------------------------------------------------------------------------------------*/
  691.  
  692. void        ZApplication::HandleAppleEvent(    AEEventClass aeClass, AEEventID aeID,
  693.                                             AppleEvent* aeEvt, AppleEvent* reply )
  694. {
  695.     if ( aeClass == kCoreEventClass )
  696.     {
  697.         FSSpec        aFile;
  698.         AEDescList    docList;
  699.         long        i, n;
  700.         AEKeyword    keyWrd;
  701.         DescType    retType;
  702.         Size        actualSize;
  703.         FInfo        fi;
  704.         
  705.         switch ( aeID )
  706.         {
  707.             case kAEReopenApplication:
  708.                 if ( GetFrontWindow())
  709.                     break;
  710.                 // fall through to normal open app event case if there are no windows:
  711.             case kAEOpenApplication:
  712.                 #if MAKE_UNTITLED_STARTUP_WINDOW
  713.                 OpenNewWindow();
  714.                 #endif
  715.                 break;
  716.             
  717.             case kAEOpenDocuments:
  718.                 FailOSErr( AEGetParamDesc( aeEvt, keyDirectObject, typeAEList, &docList ));    
  719.                 FailOSErr( AECountItems( &docList, &n ));
  720.                 
  721.                 // get each document and open it
  722.                 try
  723.                 {
  724.                     for (i = 1; i <= n; i++)
  725.                     {
  726.                         FailOSErr( AEGetNthPtr( &docList,
  727.                                                 i, 
  728.                                                 typeFSS, 
  729.                                                 &keyWrd,
  730.                                                 &retType, 
  731.                                                 &aFile, 
  732.                                                 sizeof( FSSpec ), 
  733.                                                 &actualSize ));
  734.                         
  735.                         FailOSErr( FSpGetFInfo( &aFile, &fi ));
  736.                         
  737.                         // open the file into a window and select it, provided
  738.                         // it really is one we can open. If the file is a prefs file, call up
  739.                         // the kDoPreferences command:
  740.                         
  741.                         if ( fi.fdType == 'pref' )
  742.                             HandleCommand( kCmdDoPreferences );
  743.                         else
  744.                         {
  745.                             if ( CanOpenFileType( fi.fdType ))
  746.                                 OpenFile( aFile, fi.fdType, ( fi.fdFlags & kIsStationery ) == kIsStationery );
  747.                         }
  748.                     }
  749.                 }
  750.                 catch( OSErr err )
  751.                 {
  752.                     AEDisposeDesc( &docList );
  753.                     
  754.                     throw err;
  755.                 }
  756.                 FailOSErr( AEDisposeDesc( &docList ));
  757.                 break;
  758.             
  759.             case kAEPrintDocuments:
  760.                 FailOSErr( AEGetParamDesc( aeEvt, keyDirectObject, typeAEList, &docList ));    
  761.                 FailOSErr( AECountItems( &docList, &n ));
  762.                 
  763.                 // ask the application to show the page setup dialog
  764.                 
  765.                 if ( n > 0 )
  766.                     DoPageSetup();
  767.                 
  768.                 // get each document passed by the finder
  769.                 
  770.                 try
  771.                 {
  772.                     for (i = 1; i <= n; i++)
  773.                     {
  774.                         FailOSErr( AEGetNthPtr( &docList,
  775.                                                 i, 
  776.                                                 typeFSS, 
  777.                                                 &keyWrd,
  778.                                                 &retType, 
  779.                                                 &aFile, 
  780.                                                 sizeof( FSSpec ), 
  781.                                                 &actualSize ));
  782.                         
  783.                         FailOSErr( FSpGetFInfo( &aFile, &fi ));
  784.                         
  785.                         // open the file into a window and select it
  786.                         
  787.                         if ( CanOpenFileType( fi.fdType ))
  788.                         {
  789.                             OpenFile( aFile, fi.fdType, ( fi.fdFlags & kIsStationery ) == kIsStationery );
  790.                         
  791.                             // update the window (not strictly needed, but looks better)
  792.                         
  793.                             zEH->HandleWindowUpdate( mostRecent->GetMacWindow());
  794.                         
  795.                             // ask the application to print the document
  796.                         
  797.                             DoPrint();
  798.                         
  799.                             // that done, we can close the window and move on to the next
  800.                         
  801.                             mostRecent->Close( GetPhase());
  802.                         }
  803.                     }
  804.                 }
  805.                 catch( OSErr err )
  806.                 {
  807.                     AEDisposeDesc( &docList );
  808.                     
  809.                     throw err;
  810.                 }
  811.                 
  812.                 FailOSErr( AEDisposeDesc( &docList ));
  813.                 break;
  814.             
  815.             case kAEQuitApplication:
  816.                 RequestQuit();
  817.                 break;
  818.             
  819.             default:
  820.                 ZCommander::HandleAppleEvent( aeClass, aeID, aeEvt, reply );
  821.                 break;    
  822.         }
  823.     }
  824.     else
  825.         ZCommander::HandleAppleEvent( aeClass, aeID, aeEvt, reply );
  826. }
  827.  
  828. /*-------------------------------***  HANDLEERROR  ***----------------------------------*/
  829. /*    
  830. display an alert indicating the error (some errors do nothing, likewise noErr is ignored)
  831. ----------------------------------------------------------------------------------------*/
  832.  
  833. void        ZApplication::HandleError( OSErr theErr )
  834. {
  835.     // handles the error passed by displaying an alert. This is called by the exception
  836.     // handler for the application, but you can call it at any time. Some errors are "silent"
  837.     // in that the exception does not result in a message. These include userCanceled, kSilent
  838.     // Err, and aeEventNotHandled. Override this if you want to handle errors differently.
  839.     
  840.     gMenuBar->SetTitleHilite( 0, FALSE );
  841.     
  842.     if ( theErr != userCanceledErr &&
  843.          theErr != kSilentErr       &&
  844.          theErr != noErr            &&
  845.          theErr != errAEEventNotHandled )
  846.     {
  847.         StringHandle    errExpH;    
  848.         Str255            errMsgStr;
  849.         Str31            errExplStr;
  850.         Str15            errIDStr;
  851.         
  852.         NumToString( theErr, errIDStr );
  853.         
  854.         // try to build a meaningful error message by looking for an 'Estr'
  855.         // resource with the same iD as the error. If found, this is concatenated
  856.         // onto the generic error stub and displayed. If not found, the default
  857.         // explanation "an error occurred" is used.
  858.         
  859.         GetIndString( errMsgStr, 128, 10 );
  860.         
  861.         errExpH = (StringHandle) GetResource( 'Estr', theErr );
  862.         
  863.         if ( errExpH )
  864.         {
  865.             ConcatPStrings( errMsgStr, *errExpH );     
  866.             ReleaseResource((Handle) errExpH );
  867.         }
  868.         else
  869.         {
  870.             GetIndString( errExplStr, 128, 11 );
  871.             ConcatPStrings( errMsgStr, errExplStr );
  872.         }
  873.         
  874.         ParamText( errIDStr, errMsgStr, NULL, NULL );
  875.         (void) NotifyAlert( kExceptionAlertID, ntMinimalAlert );
  876.     }
  877. }
  878.  
  879.  
  880. /*------------------------***  WAITAPPLICATIONFOREGROUND  ***---------------------------*/
  881. /*    
  882. handle events until the application is not in the background. Used by NotifyAlert().
  883. ----------------------------------------------------------------------------------------*/
  884.  
  885. void        ZApplication::WaitApplicationForeground()
  886. {
  887.     while( InBackground())
  888.         Process1Event();
  889. }
  890.  
  891.  
  892. #pragma mark -
  893.  
  894. /*------------------------------***  HANDLECOMMAND  ***---------------------------------*/
  895. /*    
  896.  
  897. Handle commands (menu items) that apply to the application as a whole, like Quit.
  898. ----------------------------------------------------------------------------------------*/
  899.  
  900. void        ZApplication::HandleCommand( const long aCmd )
  901. {
  902.     // handle commands at the application level. This includes quit, new, open, etc.
  903.     
  904.     FSSpec        aFile;
  905.     OSType        anFType;
  906.     FInfo        fi;
  907.     
  908.     switch( aCmd )
  909.     {
  910.         case kCmdAbout:
  911.             AboutBox();
  912.             break;
  913.         
  914.         case kCmdNew:
  915.             OpenNewWindowType( kDefaultWindowType );    // open a new "untitled" window
  916.             break;
  917.         
  918.         case kCmdOpen:
  919.             if ( PickFile( &aFile, &anFType ))            // choose a file
  920.             {
  921.                 // is it a stationery (template) file?
  922.                 
  923.                 FSpGetFInfo( &aFile, &fi );
  924.                 OpenFile( aFile, anFType, ( fi.fdFlags & kIsStationery ) == kIsStationery );            // open the file into a window
  925.             }
  926.             break;
  927.         
  928.         case kCmdQuit:
  929.             RequestQuit();                            // the app should now quit
  930.             break;
  931.         
  932.         case kCmdPageSetup:
  933.             DoPageSetup();
  934.             break;
  935.         
  936.         case kCmdPrint:
  937.             DoPrint();
  938.             break;
  939.         
  940.         case kCmdUndo:
  941.             if ( curUndoTask )    // undo item available?
  942.             {
  943.                 SetWatchCursor();
  944.                 if (curUndoTask->IsUndone())        // undo or redo?
  945.                     curUndoTask->Redo();            // redo the task
  946.                 else
  947.                     curUndoTask->Undo();            // undo the task
  948.             }
  949.             break;
  950.             
  951.         case kCmdDoPreferences:
  952.             DoPreferences();
  953.             break;
  954.     }
  955. }
  956.  
  957.  
  958. /*------------------------------***  HANDLECOMMAND  ***---------------------------------*/
  959.  
  960. void        ZApplication::HandleCommand( const short menuID, const short itemID )
  961. {
  962.     GrafPtr        savePort;
  963.     Str255        daName;
  964.     
  965.     if ( menuID == kAppleMenuID &&
  966.          itemID > 2 )
  967.     {
  968.         // open desk accessories.
  969.         GetMenuItemText( GetMenuHandle( menuID ), itemID, daName );
  970.         GetPort( &savePort );
  971.         OpenDeskAcc( daName );
  972.         SetPort( savePort );
  973.     }
  974. }
  975.  
  976.  
  977. /*-------------------------------***  UPDATEMENUS  ***----------------------------------*/
  978. /*    
  979.  
  980. enable the menu commands that the app can handle at the moment.
  981.  
  982. ----------------------------------------------------------------------------------------*/
  983.  
  984. void    ZApplication::UpdateMenus()
  985. {
  986.     // enable the menu commands that pertain to the application. This includes "New",
  987.     // "Quit" and "About" amongst others
  988.     
  989.     // apple menu
  990.     
  991.     gMenuBar->EnableCommand( kCmdAbout );
  992.         
  993.     // file menu
  994.     
  995.     // if memory is currently short, do not enable new or open, since those are the
  996.     // commands that are likely to allocate a lot more memory, which we do not have.
  997.     
  998.     if (! memIsShort)
  999.     {
  1000.         gMenuBar->EnableCommand( kCmdNew );
  1001.         gMenuBar->EnableCommand( kCmdOpen );
  1002.     }
  1003.     gMenuBar->EnableCommand( kCmdQuit );
  1004.         
  1005.     // enable the printing items if there is a printer and the front window
  1006.     // supports printing.
  1007.     
  1008.     if ( itsPrinter )
  1009.     {
  1010.         gMenuBar->EnableCommand( kCmdPageSetup );
  1011.     
  1012.         ZWindow*    fWindow = GetFrontWindow();
  1013.         
  1014.         if (fWindow && fWindow->IsPrintable())
  1015.             gMenuBar->EnableCommand( kCmdPrint );
  1016.     }    
  1017.     
  1018.     // edit:
  1019.     
  1020.     UpdateUndo();
  1021.     gMenuBar->EnableCommand( kCmdDoPreferences );
  1022. }
  1023.  
  1024.  
  1025. /*-------------------------------***  DOPAGESETUP  ***----------------------------------*/
  1026. /*
  1027. handle the Page Setup command    
  1028. ----------------------------------------------------------------------------------------*/
  1029.  
  1030. void    ZApplication::DoPageSetup()
  1031. {
  1032.     if ( itsPrinter )
  1033.     {
  1034.         gWindowManager->Deactivate();
  1035.         itsPrinter->PageSetUp();
  1036.         gWindowManager->Activate();
  1037.     }
  1038. }
  1039.  
  1040. /*----------------------------------***  DOPRINT  ***-----------------------------------*/
  1041. /*
  1042. handle the Print command    
  1043. ----------------------------------------------------------------------------------------*/
  1044.  
  1045. void    ZApplication::DoPrint()
  1046. {
  1047.     if ( itsPrinter )
  1048.     {
  1049.         ZWindow*    aWindow = GetFrontWindow();
  1050.         
  1051.         gWindowManager->Deactivate();
  1052.         itsPrinter->Print( aWindow );
  1053.         gWindowManager->Activate();
  1054.     }
  1055. }
  1056.  
  1057.  
  1058. /*---------------------------------***  ABOUTBOX  ***-----------------------------------*/
  1059. /*    
  1060. display the application's about box
  1061. ----------------------------------------------------------------------------------------*/
  1062.  
  1063. void    ZApplication::AboutBox()
  1064. {
  1065.     gWindowManager->DeactivateForDialog( kAboutBoxID, TRUE );
  1066.     (void) Alert( kAboutBoxID, NULL );
  1067.     gWindowManager->Activate();
  1068. }
  1069.  
  1070.  
  1071. /*----------------------------------***  SETTASK  ***-----------------------------------*/
  1072. /*    
  1073. update the current global undo task
  1074. ----------------------------------------------------------------------------------------*/
  1075.  
  1076. void    ZApplication::SetTask( ZUndoTask* aTask )
  1077. {
  1078.     if ( curUndoTask )
  1079.         ForgetObject( curUndoTask );
  1080.         
  1081.     curUndoTask = aTask;
  1082. }
  1083.  
  1084. /*--------------------------------***  UPDATEUNDO  ***----------------------------------*/
  1085. /*    
  1086. update the Undo menu item to reflect the current undo task
  1087. ----------------------------------------------------------------------------------------*/
  1088.  
  1089. void    ZApplication::UpdateUndo()
  1090. {
  1091.     Str255        undoMenuStr;
  1092.     
  1093.     if ( curUndoTask )
  1094.     {
  1095.         // a task available, so update the undo menu item to the task string.
  1096.         
  1097.         Str63        taskStr;
  1098.         
  1099.         if (curUndoTask->IsUndone())
  1100.             GetIndString( undoMenuStr, kMiscStrListID, kRedoStrIndex );
  1101.         else
  1102.             GetIndString( undoMenuStr, kMiscStrListID, kUndoStrIndex );
  1103.             
  1104.         // add the task name
  1105.         
  1106.         curUndoTask->GetTaskString( taskStr );
  1107.         ConcatPStrings( undoMenuStr, taskStr );
  1108.         
  1109.         // set the menu item to the task name
  1110.         
  1111.         gMenuBar->SetCommandText( kCmdUndo, undoMenuStr );
  1112.         
  1113.         // there is a task, but does it refer to the front window? If not,
  1114.         // then do not actually enable the command.
  1115.  
  1116.         if ((curUndoTask->GetUndoTarget() == GetFrontWindow()) &&
  1117.             (curUndoTask->GetUndoTarget() != NULL))
  1118.             gMenuBar->EnableCommand( kCmdUndo );
  1119.     }
  1120.     else
  1121.     {
  1122.         // if no task, just show a dimmed "Can't Undo"
  1123.         
  1124.         gMenuBar->SetCommandText( kCmdUndo, kMiscStrListID, kCantUndoStrIndex );
  1125.     }
  1126. }
  1127.  
  1128.  
  1129. #pragma mark -
  1130. /*-------------------------------***  INITMENUBAR  ***----------------------------------*/
  1131. /*    
  1132. set up the menubar object for managing the main menus
  1133. ----------------------------------------------------------------------------------------*/
  1134.  
  1135. void        ZApplication::InitMenuBar()
  1136. {
  1137.     // installs the menu bar. By default, we just install 'MBAR' ID = 128, which means you
  1138.     // don't need to override this to get other menus- just create the resources you want.
  1139.     
  1140.     FailNIL( gMenuBar = new ZMenuBar( kStdMenubarID ));
  1141.  
  1142.     gMenuBar->InitMenuBar();
  1143. }
  1144.  
  1145.  
  1146. /*---------------------------***  MOUSENOTINANYWINDOW  ***------------------------------*/
  1147. /*    
  1148. the mouse is not over any window. By default this just sets the default cursor shape, but
  1149. you can override this to get informed if you are interested in this fact.
  1150. ----------------------------------------------------------------------------------------*/
  1151.  
  1152. void    ZApplication::MouseNotInAnyWindow( const Point globalMouse )
  1153. {
  1154.     ResumeCursorAnimation();
  1155.     SetCursorShape( 0 );
  1156. }
  1157.  
  1158.  
  1159. /*--------------------------------***  MAKECLIPBOARD  ***-------------------------------*/
  1160. /*    
  1161.  
  1162. Make the clipboard object. By default, this is a ZClipboard.
  1163. ----------------------------------------------------------------------------------------*/
  1164.  
  1165. void        ZApplication::MakeClipboard()
  1166. {
  1167.     FailNIL( gClipboard = new ZClipboard());
  1168. }
  1169.  
  1170.  
  1171. /*------------------------------***  MAKEEVENTHANDLER  ***------------------------------*/
  1172. /*    
  1173.  
  1174. Make the event handler object. By default, this is a ZEventHandler. This also makes the
  1175. window manager, since that is functionally related to the event handler.
  1176. ----------------------------------------------------------------------------------------*/
  1177.  
  1178.  
  1179. void        ZApplication::MakeEventHandler()
  1180. {
  1181.     // make the event handler
  1182.     
  1183.     FailNIL( zEH = new ZEventHandler());
  1184.     
  1185.     // make the window manager for handling floating windows, etc.
  1186.     
  1187.     FailNIL( gWindowManager = new ZWindowManager());
  1188.     
  1189.     // install handlers for the four required events
  1190.     
  1191.     zEH->InstallApplescriptHandlers();        
  1192. }
  1193.  
  1194.  
  1195. /*--------------------------------***  MAKEPRINTER  ***---------------------------------*/
  1196. /*    
  1197.  
  1198. creates a new printer object for handling the print commands
  1199.  
  1200. ----------------------------------------------------------------------------------------*/
  1201.  
  1202. void        ZApplication::MakePrinter()
  1203. {
  1204.     #if PRINTING_ON
  1205.     
  1206.     try
  1207.     {
  1208.         FailNIL( itsPrinter = new ZPrinter());
  1209.     }
  1210.     catch( OSErr err )
  1211.     {
  1212.         itsPrinter = NULL;
  1213.         
  1214.         // note, to not have a printer is not fatal- in this case we carry on,
  1215.         // but print commands will be greyed out. This may happen if no printer
  1216.         // driver is selected in the chooser.
  1217.     }
  1218.     #endif
  1219. }
  1220.  
  1221. #pragma mark -
  1222.  
  1223. /*------------------------------***  MAKENEWWINDOW  ***---------------------------------*/
  1224. /*    
  1225.  
  1226. creates a new window object and initialises it. It does not show it yet. Normally you will
  1227. override this method to create your own useful kinds of windows. However, even if you don't
  1228. you'll still get a window that works. Eat your heart out Sprocket!
  1229. ----------------------------------------------------------------------------------------*/
  1230.  
  1231. void        ZApplication::MakeNewWindow()
  1232. {
  1233.     FailNIL( mostRecent = new ZWindow( this,  kUntitledWindowID ));
  1234.     
  1235.     try
  1236.     {
  1237.         mostRecent->InitZWindow();
  1238.     }
  1239.     catch( OSErr err )
  1240.     {
  1241.         ForgetObject( mostRecent );
  1242.         
  1243.         throw err;
  1244.     }
  1245. }
  1246.  
  1247.  
  1248. /*------------------------------***  OPENNEWWINDOW  ***---------------------------------*/
  1249. /*    
  1250.  
  1251. creates a new window object and initialises it. It then shows it and makes it active.
  1252.  
  1253. ----------------------------------------------------------------------------------------*/
  1254.  
  1255. void        ZApplication::OpenNewWindow()
  1256. {
  1257.     ZWindow*    theWindow;
  1258.     
  1259.     gMenuBar->SetZoomSourceToCommand( kCmdNew );
  1260.     
  1261.     theWindow = mostRecent = MakeNewWindowType( kDefaultWindowType );
  1262.     
  1263.     if ( theWindow )
  1264.     {
  1265.         theWindow->Place();
  1266.         theWindow->Select();
  1267.     }
  1268. }
  1269.  
  1270.  
  1271.  
  1272. /*----------------------------***  OPENNEWWINDOWTYPE  ***-------------------------------*/
  1273. /*    
  1274. create a new window object of the requisite type. This is the modern version of
  1275. OpenNewWindow(), but older code will work as expected.
  1276. ----------------------------------------------------------------------------------------*/
  1277.  
  1278. ZWindow*    ZApplication::OpenNewWindowType( OSType aType )
  1279. {
  1280.     ZWindow*    theWindow;
  1281.     
  1282.     gMenuBar->SetZoomSourceToCommand( kCmdNew );
  1283.  
  1284.     theWindow = mostRecent = MakeNewWindowType( aType );
  1285.     
  1286.     if ( theWindow )
  1287.     {
  1288.         theWindow->Place();
  1289.         theWindow->Select();
  1290.     }
  1291.     
  1292.     return theWindow;
  1293. }
  1294.  
  1295.  
  1296. /*---------------------------------***  CLOSEALL  ***-----------------------------------*/
  1297. /*    
  1298.  
  1299. closes all of the application's windows. If one refuses to close, this will abort a Quit.
  1300. This iterates backwards through its underlings list looking for windows, rather than
  1301. relying on the window lists, since windows will also close their own underling windows so
  1302. the whole commander tree is cleaned up properly. If your app organises things differently,
  1303. you may need to override this.
  1304. ----------------------------------------------------------------------------------------*/
  1305.  
  1306. void    ZApplication::CloseAll( Boolean closeFloaters )
  1307. {
  1308.     long        i;
  1309.     ZWindow*    aWindow;
  1310.     
  1311.     if ( itsUnderlings )
  1312.     {
  1313.         i = itsUnderlings->CountItems();
  1314.         
  1315.         while( i )
  1316.         {
  1317.             aWindow = dynamic_cast<ZWindow*>( itsUnderlings->GetObject( i-- ));
  1318.             
  1319.             if ( aWindow )
  1320.             { 
  1321.                 if ( phase == kQuitting )
  1322.                 {
  1323.                     if ( ! aWindow->Close( phase ))
  1324.                     {
  1325.                         phase = kRunning;
  1326.                         done = FALSE;
  1327.                         break;
  1328.                     }
  1329.                 }
  1330.                 else
  1331.                 {
  1332.                     if (! aWindow->NoAutoClose() && ( aWindow->Floats() == closeFloaters ))
  1333.                     {
  1334.                         if ( ! aWindow->Close( phase ))
  1335.                         {
  1336.                             phase = kRunning;
  1337.                             done = FALSE;
  1338.                             break;
  1339.                         }
  1340.                     }
  1341.                 }
  1342.             }
  1343.         }
  1344.     }
  1345. }
  1346.  
  1347. /*------------------------------***  GETFRONTWINDOW  ***--------------------------------*/
  1348. /*    
  1349.  
  1350. returns the active window object, if there is one
  1351.  
  1352. ----------------------------------------------------------------------------------------*/
  1353.  
  1354. ZWindow*    ZApplication::GetFrontWindow()
  1355. {
  1356.     return ( gWindowManager->GetTopWindow());
  1357. }
  1358.  
  1359.  
  1360. /*----------------------------------***  PICKFILE  ***----------------------------------*/
  1361. /*    
  1362.  
  1363. displays the standard file dialog for selecting a file, and returns its filespec.
  1364.  
  1365. ----------------------------------------------------------------------------------------*/
  1366.  
  1367.  
  1368. Boolean        ZApplication::PickFile( FSSpec* aFile, OSType* fType )
  1369. {
  1370.     // uses standard file to choose a file to open to a window. By default, no files types
  1371.     // are added to the list, which we here interpret to mean "show all files".
  1372.  
  1373. #if _USE_NAVIGATION_SERVICES
  1374.     // using navigation services, so we need to tackle things a little differently. The result
  1375.     // from this method is the same- namely a single filespec and a type.
  1376.     
  1377.     if ( gMacInfo.hasNavigationServices )
  1378.     {
  1379.         OSErr                theErr;
  1380.         NavReplyRecord        navReply;
  1381.         NavDialogOptions    navOptions;
  1382.         Boolean                result = FALSE;
  1383.         
  1384.         FailOSErr( NavGetDefaultDialogOptions( &navOptions ));
  1385.         
  1386.         GetName( navOptions.clientName );
  1387.         
  1388.         // only permit one selection here:
  1389.         
  1390.         navOptions.dialogOptionFlags &= ~kNavAllowMultipleFiles;
  1391.         gWindowManager->Deactivate();
  1392.         StopCursorAnimation();
  1393.         
  1394.         theErr = NavGetFile( NULL,
  1395.                              &navReply,
  1396.                              &navOptions,
  1397.                              gNavEventHandler,
  1398.                              NULL,
  1399.                              NULL,
  1400.                              (NavTypeListHandle) itsFileTypes,
  1401.                              (NavCallBackUserData) this );
  1402.                              
  1403.         gWindowManager->Activate();
  1404.                                 
  1405.         // extract info:
  1406.         
  1407.         if (( theErr == noErr ) && navReply.validRecord )
  1408.         {
  1409.             long    count, i;
  1410.             AEDesc    specDesc;
  1411.             FInfo    fi;
  1412.             
  1413.             FailOSErr( AECountItems( &navReply.selection, &count ));
  1414.             
  1415.             for( i = 1; i <= count; i++ )
  1416.             {
  1417.                 FailOSErr( AEGetNthDesc( &navReply.selection, i, typeFSS, NULL, &specDesc ));
  1418.             
  1419.                 BlockMoveData( *specDesc.dataHandle, aFile, sizeof( FSSpec ));
  1420.                 
  1421.                 // to get the type we need to do a FsGetFInfo:
  1422.                 
  1423.                 FSpGetFInfo( aFile, &fi );
  1424.                 *fType = fi.fdType;
  1425.             }
  1426.         
  1427.         
  1428.             result = TRUE;
  1429.         }
  1430.         else
  1431.             result = FALSE;
  1432.             
  1433.         FailOSErr( NavDisposeReply( &navReply ));
  1434.         
  1435.         return result;
  1436.     }
  1437.     else
  1438.     {
  1439. #endif
  1440.  
  1441.     StandardFileReply    aReply;
  1442.     OSType*                listPtr;
  1443.     short                numTypes;
  1444.     
  1445.     HLock((Handle) itsFileTypes );
  1446.     numTypes = (*itsFileTypes )->osTypeCount;
  1447.     listPtr = &(*itsFileTypes)->osType[0];
  1448.     
  1449.     // if no types in the list, show all of them
  1450.     
  1451.     if( numTypes <= 0 )
  1452.         numTypes = -1;
  1453.     
  1454.     StopCursorAnimation();
  1455.     gWindowManager->DeactivateForDialog( sfGetDialogID );
  1456.     
  1457.     // display the dialog
  1458.     StandardGetFile( NULL, numTypes, listPtr, &aReply );
  1459.     HUnlock((Handle) itsFileTypes );
  1460.     
  1461.     gWindowManager->Activate();
  1462.     
  1463.     if ( aReply.sfGood )
  1464.     {
  1465.         *aFile = aReply.sfFile;
  1466.         *fType = aReply.sfType;
  1467.         return TRUE;
  1468.     }
  1469.     else
  1470.         return FALSE;
  1471.  
  1472. #if _USE_NAVIGATION_SERVICES
  1473.     }
  1474. #endif
  1475. }
  1476.  
  1477.  
  1478. /*----------------------------------***  OPENFILE  ***----------------------------------*/
  1479. /*    
  1480.  
  1481. creates a new window and asks it to open the file. It then shows the window and activates it.
  1482. If your application doesn't have a classic "document" interface, but e.g. simply processes
  1483. files dropped on the application, you can override this method to process the files. It will
  1484. be called once for each file dropped on the application that we know how to open.
  1485. ----------------------------------------------------------------------------------------*/
  1486.  
  1487. void        ZApplication::OpenFile( const FSSpec& aFile, const OSType fType, Boolean isStationery )
  1488. {
  1489.     // opens the file into a new window. This is equivalent to OpenNewWindow, but
  1490.     // for when the user chose a file with the Open command
  1491.     
  1492.     ZWindow*    aWindow = NULL;
  1493.     
  1494.     SetWatchCursor();
  1495.     aWindow = mostRecent = MakeNewWindowType( fType );
  1496.     
  1497.     // window created, so ask it to open the chosen file
  1498.     
  1499.     if ( aWindow )
  1500.     {
  1501.         try
  1502.         {
  1503.             aWindow->SetFile( aFile );
  1504.             aWindow->OpenFile( fType, isStationery );
  1505.             aWindow->Place();
  1506.             aWindow->Select();
  1507.         }
  1508.         catch( OSErr err )
  1509.         {
  1510.             ForgetObject( aWindow );
  1511.             throw err;
  1512.         }
  1513.     }
  1514. }
  1515.  
  1516.  
  1517. /*--------------------------------***  ADDFILETYPE  ***---------------------------------*/
  1518. /*    
  1519. adds <aType> to the list of types this application will show in the Open dialog. You can
  1520. call this for each type your application can open. This ignores duplicates.
  1521. -----------------------------------------------------------------------------------------*/
  1522.  
  1523. void        ZApplication::AddFileType( const OSType aType )
  1524. {
  1525.     // adds the file type to the list of types, if not already there
  1526.     
  1527.     short    nTypes, i;
  1528.     
  1529.     nTypes = (*itsFileTypes)->osTypeCount;
  1530.     
  1531.     // check that the file type we are adding is unique in the list
  1532.     
  1533.     for ( i = 0; i < nTypes; i++ )
  1534.     {
  1535.         if ((*itsFileTypes)->osType[i] == aType )
  1536.             return;
  1537.     }
  1538.     
  1539.     // if we are still here, type is unique, so append it
  1540.     // first grow the handle
  1541.     
  1542.     SetHandleSize((Handle) itsFileTypes, GetHandleSize((Handle) itsFileTypes ) + sizeof( OSType ));
  1543.     FailMemError();
  1544.     
  1545.     // set the new entry
  1546.     
  1547.     (*itsFileTypes)->osType[ nTypes ] = aType;
  1548.     (*itsFileTypes)->osTypeCount++;
  1549. }
  1550.  
  1551.  
  1552. /*------------------------------***  CANOPENFILETYPE  ***--------------------------------*/
  1553. /*
  1554. return TRUE if this application can open files of the given type. The default method
  1555. determines if the type is in the fileType list or not. If the list is empty, this always
  1556. return TRUE.    
  1557. -----------------------------------------------------------------------------------------*/
  1558.  
  1559. Boolean        ZApplication::CanOpenFileType( const OSType aType )
  1560. {
  1561.     short    nTypes, i;
  1562.     
  1563.     nTypes = (*itsFileTypes)->osTypeCount;
  1564.     
  1565.     if ( nTypes > 0 )
  1566.     {
  1567.         for ( i = 0; i < nTypes; i++ )
  1568.         {
  1569.             if ( (*itsFileTypes)->osType[i] == aType )
  1570.                 return TRUE;
  1571.         }
  1572.         
  1573.         return FALSE;
  1574.     }
  1575.     else
  1576.         return TRUE;
  1577. }
  1578.  
  1579.  
  1580. /*------------------------------***  REGISTERCLASSES  ***-------------------------------*/
  1581. /*    
  1582. in order to use persistent objects, classes must be registered. You only need to register
  1583. classes that you will want to create from a stream. By default this registers those
  1584. required classes that could be constructed from a stream- override this to extend the
  1585. range of classes, but always call the inherited method.
  1586. ----------------------------------------------------------------------------------------*/
  1587.  
  1588. void        ZApplication::RegisterClasses()
  1589. {
  1590. #if _MACZOOP_STREAMS
  1591.  
  1592.     FailNIL( gClasses );
  1593.     
  1594.     REGISTERCLASS( ZComrade );
  1595.     REGISTERCLASS( ZCommander );
  1596.     REGISTERCLASS( ZArray );
  1597.     REGISTERCLASS( ZWindow );
  1598.     REGISTERCLASS( ZApplication );
  1599.     REGISTERCLASS( ZObjectList );
  1600.  
  1601. #endif
  1602. }
  1603.  
  1604.  
  1605.  
  1606. #pragma mark -
  1607. /*---------------------------------***  ZGROWFUNC  ***----------------------------------*/
  1608. /*    
  1609.  
  1610. memory manager callback proc.
  1611. ----------------------------------------------------------------------------------------*/
  1612.  
  1613. static pascal long ZGrowFunc( Size bytesShort )
  1614. {
  1615.     Boolean bytesFreed;
  1616.     Size    growBytes;
  1617.     
  1618.     // call application object to free some memory
  1619.     try
  1620.     {
  1621.         bytesFreed = gApplication->MemoryShortage( bytesShort );
  1622.         
  1623.         // try to compact and purge the memory after the
  1624.         // application has done something to free some up.
  1625.         
  1626.         (void) MaxMem( &growBytes );
  1627.     }
  1628.     catch( OSErr err )
  1629.     {
  1630.         bytesFreed = 0;
  1631.     }
  1632.     
  1633.     return bytesFreed;
  1634. }
  1635.  
  1636.  
  1637. /******************************************************************************
  1638.  CopyPString
  1639.  
  1640.         Copy a Pascal string
  1641.  ******************************************************************************/
  1642.  
  1643. void    CopyPString( ConstStr255Param srcString, Str255 destString )
  1644. {
  1645.     BlockMoveData( srcString, destString, MIN( srcString[0] + 1, 255 ));
  1646. }
  1647.  
  1648.  
  1649. /******************************************************************************
  1650.  ConcatPStrings
  1651.  
  1652.         Concatenate two Pascal strings by attaching the second string on
  1653.         the end of the first string.
  1654.  ******************************************************************************/
  1655.  
  1656. void    ConcatPStrings( Str255 root, ConstStr255Param append )
  1657. {
  1658.     short charsToCopy;
  1659.  
  1660.     // Truncate if concatenated string would be longer than 255 chars.
  1661.  
  1662.     charsToCopy = MIN( append[0], 255 - root[0]);
  1663.     BlockMoveData( append + 1, root + root[0] + 1, (long) charsToCopy);
  1664.     root[0] += charsToCopy;
  1665. }
  1666.  
  1667.  
  1668. /******************************************************************************
  1669.  CopyPStringTrunc
  1670.  
  1671.         Copy a Pascal string, limiting length to <ccLim> characters. This is
  1672.         handy for copying a Str255 to a shorter type, e.g. Str31.
  1673.  ******************************************************************************/
  1674.  
  1675. void    CopyPStringTrunc( ConstStr255Param srcString, Str255 destString, unsigned char ccLim )
  1676. {
  1677.     BlockMoveData( srcString, destString, MIN( srcString[0] + 1, ccLim + 1 ));
  1678.     destString[0] = MIN( srcString[0], ccLim );
  1679. }
  1680.  
  1681.  
  1682. /******************************************************************************
  1683.  ConcatPStringsTrunc
  1684.  
  1685.         Concatenate two Pascal strings by attaching the second string on
  1686.         the end of the first string, but limiting the overall length to <ccLim>
  1687.  ******************************************************************************/
  1688.  
  1689. void    ConcatPStringsTrunc( Str255 root, ConstStr255Param append, unsigned char ccLim )
  1690. {
  1691.     short charsToCopy;
  1692.  
  1693.     // Truncate if concatenated string would be longer than <ccLim> chars.
  1694.  
  1695.     charsToCopy = MIN( append[0], ccLim - root[0]);
  1696.     BlockMoveData( append + 1, root + root[0] + 1, (long) charsToCopy);
  1697.     root[0] += charsToCopy;
  1698. }
  1699.  
  1700.  
  1701.  
  1702. /*------------------------------***  RUNAPPLICATION  ***--------------------------------*/
  1703. /*    
  1704. standard function to run the application. Your main() function must make the relevant
  1705. ZApplication object, assign it to gApplication, then call this.
  1706. ----------------------------------------------------------------------------------------*/
  1707.  
  1708. void    RunApplication()
  1709. {
  1710.     if (gApplication)
  1711.     {    
  1712.         try
  1713.         {
  1714.             gApplication->InitMacZoop();            // initialise the whole kaboodle. This is
  1715.                                                     // NOT done by the constructor since you might
  1716.                                                     // want to override the initialisation.
  1717.         }
  1718.         catch( OSErr err )
  1719.         {
  1720.             // if an exception occurs during startup, the application cannot run, since
  1721.             // everything must be properly built and in place before handling events. In this
  1722.             // case we display a fatal alert message and exit.
  1723.             Str31    appName, errStr;
  1724.             
  1725.             gApplication->GetName( appName );
  1726.             NumToString( err, errStr );
  1727.             ParamText( appName, errStr, NULL, NULL );
  1728.         
  1729.             (void) Alert( kFatalStartupErrAlertID, NULL );
  1730.             ExitToShell();
  1731.         }
  1732.         
  1733.         // initialisation is now complete, so we can go ahead and run the thing
  1734.         
  1735.         do
  1736.         {
  1737.             gApplication->Run();                    // run the application until the user quits
  1738.         }
  1739.         while (! gApplication->Quit());                // try to quit
  1740.     }
  1741. }
  1742.  
  1743.  
  1744.  
  1745. Boolean    IsColourPort( GrafPtr aPort )
  1746. {
  1747.     return (( aPort->portBits.rowBytes & 0x8000 ) != 0 );
  1748. }
  1749.  
  1750.  
  1751. void    SetHiliteMode()
  1752. {
  1753.     LMSetHiliteMode( LMGetHiliteMode() & pHiliteBit );
  1754. }
  1755.  
  1756.  
  1757. /*----------------------------------***  MACHASDM  ***----------------------------------*/
  1758. /*    
  1759.  
  1760. static function returns TRUE if Drag Manager is present. Checks link on PowerMacs.
  1761. ----------------------------------------------------------------------------------------*/
  1762.  
  1763. Boolean        MacHasDM()
  1764. {
  1765.     // returns TRUE if drag manager is installed on this mac and is loaded by the CFM.
  1766.     
  1767.     Boolean hasDM = gMacInfo.hasDragManager;
  1768.     
  1769.     #if GENERATINGCFM 
  1770.         // check that the dragLib is actually loaded and linked
  1771.         
  1772.         hasDM = hasDM && ( NewDrag != (void*) kUnresolvedCFragSymbolAddress );
  1773.  
  1774.     #endif
  1775.  
  1776.     return hasDM;
  1777. }
  1778.  
  1779.  
  1780.  
  1781. /*--------------------------------***  NOTIFYALERT  ***---------------------------------*/
  1782. /*    
  1783. Works just like Alert(), except that if the app is in the background, the notification
  1784. manager is used to inform the user that the app requires attention.
  1785. ----------------------------------------------------------------------------------------*/
  1786.  
  1787. static NMRec    gNotification;
  1788. static Str255    gNotifyMessage;
  1789. static Boolean    gNotificationPosted = FALSE;
  1790.  
  1791. short        NotifyAlert( const short alertID, NTAlertFlags ntFlags )
  1792. {
  1793.     // basically: if ( background ):
  1794.     //                install notification
  1795.     //                wait for app foreground
  1796.     //                delete notification
  1797.     
  1798.     if ( ! gNotificationPosted )
  1799.     {
  1800.         if ( gApplication->InBackground())
  1801.         {
  1802.             // if an alert is to be displayed, set it up
  1803.             
  1804.             if ( ntFlags & ntAlertDisplayMessage )
  1805.             {
  1806.                 // build a string that says "The application “<name>” requires your attention.
  1807.                 // Please bring it to the front.
  1808.                 
  1809.                 Str255    appName;
  1810.                 
  1811.                 GetIndString( gNotifyMessage, kMiscStrListID, 17 );
  1812.                 gApplication->GetName( appName );
  1813.                 ConcatPStrings( gNotifyMessage, appName );
  1814.                 GetIndString( appName, kMiscStrListID, 18 );
  1815.                 ConcatPStrings( gNotifyMessage, appName );
  1816.                 
  1817.                 gNotification.nmStr = gNotifyMessage;
  1818.             }
  1819.             else
  1820.                 gNotification.nmStr = NULL;
  1821.             
  1822.             // set up any sound:
  1823.             
  1824.             if ( ntFlags & ntAlertPlaySound )
  1825.                 gNotification.nmSound = ( Handle ) -1L;
  1826.             else
  1827.                 gNotification.nmSound = NULL;
  1828.             
  1829.             Handle    appIconSuiteH;
  1830.             OSErr    theErr;
  1831.                 
  1832.             // set up icon and menu mark
  1833.             
  1834.             gNotification.qType = 8;
  1835.             gNotification.nmMark = TRUE;
  1836.             gNotification.nmResp = NULL;
  1837.             gNotification.nmRefCon = NULL;
  1838.             
  1839.             theErr = GetIconSuite( &appIconSuiteH, kApplicationIconSuiteID, svAllAvailableData );
  1840.             
  1841.             if ( theErr == noErr )
  1842.                 gNotification.nmIcon = appIconSuiteH;
  1843.             else
  1844.                 gNotification.nmIcon = NULL;
  1845.                 
  1846.             // install notification:
  1847.  
  1848.             NMInstall( &gNotification );
  1849.                 
  1850.             gNotificationPosted = TRUE;
  1851.             
  1852.             // handle events until we come back to the front:
  1853.         
  1854.             gApplication->WaitApplicationForeground();
  1855.             
  1856.             // we're back, so delete the notification
  1857.             
  1858.             NMRemove( &gNotification );
  1859.             gNotificationPosted = FALSE;
  1860.             
  1861.             if ( appIconSuiteH )
  1862.                 DisposeIconSuite( appIconSuiteH, FALSE );
  1863.         }
  1864.  
  1865.         return Alert( alertID, NULL );
  1866.     }
  1867.     else
  1868.     {
  1869.         // what can we do? This shouldn't arise in normal use, but just in case, we act as
  1870.         // though the user saw an alert and clicked OK. Not ideal, but since a notification
  1871.         // can't interrupt another, there's little choice.
  1872.         
  1873.         SysBeep( 1 );
  1874.         return ok;
  1875.     }
  1876. }
  1877.  
  1878. // header invariant version of Delay():
  1879.  
  1880. void    MZDelay( short ticks )
  1881. {
  1882.     #ifdef __COMPATIBILITY__
  1883.         long ignored;
  1884.     #else
  1885.         unsigned long ignored;
  1886.     #endif
  1887.     
  1888.     Delay( ticks, &ignored );
  1889. }
  1890.  
  1891.  
  1892.  
  1893. void    MZWait( unsigned short ticks )
  1894. {
  1895.     long    tc = TickCount() + ticks;
  1896.     
  1897.     while( TickCount() < tc )
  1898.         gApplication->Process1Event();
  1899. }
  1900.  
  1901.  
  1902.  
  1903. void    AssertErr( long lineNo, char* srcFile, char* reason, long val )
  1904. {
  1905. #if _DEBUG_
  1906.     
  1907.     Str32        lineStr, valStr;
  1908.     StringPtr    fileStr, reasonStr;
  1909.     
  1910.     NumToString( lineNo, lineStr );
  1911.     NumToString( val, valStr );
  1912.     fileStr = c2pstr( srcFile );
  1913.     reasonStr = c2pstr( reason );
  1914.     
  1915.     ParamText( lineStr, fileStr, reasonStr, valStr );
  1916.     StopCursorAnimation();
  1917.     
  1918.     (void) Alert( kAssertionAlertID, NULL );
  1919.     
  1920.     FailOSErr( kSilentErr );
  1921.     
  1922. #endif
  1923. }
  1924.  
  1925.  
  1926. #if _USE_NAVIGATION_SERVICES
  1927. pascal void    ZNavEventCallback(     NavEventCallbackMessage cbMessage,
  1928.                                 NavCBRecPtr    cbParams,
  1929.                                 NavCallBackUserData    cbData )
  1930. {
  1931.     switch ( cbMessage )
  1932.     {
  1933.         case kNavCBEvent:
  1934.             // we are only interested in updates actually
  1935.             
  1936.             if ( cbParams->eventData.event->what == updateEvt )
  1937.                 gApplication->Process1Event( cbParams->eventData.event );
  1938.             break;
  1939.             
  1940.         default:
  1941.             break;
  1942.     
  1943.     
  1944.     }
  1945. }
  1946.  
  1947. #endif
  1948.